阅读目录
一、MyBatis-Plus
1、简介
2、使用 SpringBoot 快速使用 MyBatis-Plus
二、Mybatis-Plus 常用操作
1、配置日志
2、简单认识一下常用注解
3、代码生成器
4、自动填充数据功能
5、逻辑删除
6、分页插件的使用
7、乐观锁的实现
三、Mybatis-Plus CRUD 操作简单了解一下
1、Mapper 接口方法(CRUD)简单了解一下
2、Service 接口方法(CRUD)简单了解一下
3、条件构造器(Wrapper,定义 where 条件)
回到顶部
一、MyBatis-Plus
1、简介
MyBatis-Plus 是一个 Mybatis 增强版工具,在 MyBatis 上扩充了其他功能没有改变其基本功能,为了简化开发提交效率而存在。
官网文档地址: https://mp.baomidou.com/guide/
MyBatis-Plus 特性: https://mp.baomidou.com/guide/#%E7%89%B9%E6%80%A7
2、使用 SpringBoot 快速使用 MyBatis-Plus
(1)准备工作 需要 Java 开发环境(JDK)以及相应的开发工具(IDE)。 需要 maven(用来下载相关依赖的 jar 包)。 需要 SpringBoot。 可以使用 IDEA 安装一个 mybatis-plus 插件。
![](https://img2020.cnblogs.com/blog/1688578/202005/1688578-20200509182324596-297575739.png)
(2)创建一个 SpringBoot 项目。 方式一:去官网 https://start.spring.io/ 初始化一个,然后导入 IDE 工具即可。 方式二:直接使用 IDE 工具创建一个。 Spring Initializer。
![](https://img2020.cnblogs.com/blog/1688578/202005/1688578-20200509182351692-1490253732.png)
![](https://img2020.cnblogs.com/blog/1688578/202005/1688578-20200509182402784-1080563045.png)
(3)添加 MyBatis-Plus 依赖(mybatis-plus-boot-starter)
com.baomidou
mybatis-plus-boot-starter
3.3.1.tmp
(4)为了测试开发,此处使用 mysql 8,需要引入 mysql 相关依赖。 为了简化代码,引入 lombok 依赖(减少 getter、setter 等方法)。
mysql
mysql-connector-java
8.0.18
org.projectlombok
lombok
1.18.10
(5)完整依赖文件(pom.xml)
4.0.0
org.springframework.boot
spring-boot-starter-parent
2.2.6.RELEASE
com.lyh.test
test-mybatis-plus
0.0.1-SNAPSHOT
test-mybatis-plus
测试 -- 测试 MyBatis-Plus 功能
1.8
com.baomidou
mybatis-plus-boot-starter
3.3.1.tmp
mysql
mysql-connector-java
8.0.18
org.projectlombok
lombok
1.18.10
org.springframework.boot
spring-boot-starter
org.springframework.boot
spring-boot-starter-test
test
org.junit.vintage
junit-vintage-engine
org.springframework.boot
spring-boot-maven-plugin
(6)使用一个表进行测试。
仅供参考,可以定义 创建时间、修改时间等字段。
DROP DATABASE IF EXISTS testMyBatisPlus;
CREATE DATABASE testMyBatisPlus;
USE testMyBatisPlus;
DROP TABLE IF EXISTS user;
CREATE TABLE user
(
id BIGINT(20) NOT NULL COMMENT '主键ID',
name VARCHAR(30) NULL DEFAULT NULL COMMENT '姓名',
age INT(11) NULL DEFAULT NULL COMMENT '年龄',
email VARCHAR(50) NULL DEFAULT NULL COMMENT '邮箱',
PRIMARY KEY (id)
);
INSERT INTO user (id, name, age, email) VALUES
(1, 'Jone', 18, '[email protected]'),
(2, 'Jack', 20, '[email protected]'),
(3, 'Tom', 28, '[email protected]'),
(4, 'Sandy', 21, '[email protected]'),
(5, 'Billie', 24, '[email protected]');
![](https://img2020.cnblogs.com/blog/1688578/202005/1688578-20200509182613102-1729540101.png)
(7)在 application.yml 文件中配置 mysql 数据源信息。
spring:
datasource:
driver-class-name: com.mysql.cj.jdbc.Driver
username: root
password: 123456
url: jdbc:mysql://localhost:3306/testMyBatisPlus?useUnicode=true&characterEncoding=utf8
![](https://img2020.cnblogs.com/blog/1688578/202005/1688578-20200509182640635-986081965.png)
(8)编写表对应的 实体类。
package entity;
import lombok.Data;
@Data
public class User {
private Long id;
private String name;
private int age;
private String email;
}
![](https://img2020.cnblogs.com/blog/1688578/202005/1688578-20200509182707965-401919746.png)
(9)编写操作实体类的 Mapper 类。 直接继承 BaseMapper,这是 mybatis-plus 封装好的类。
package mapper;
import bean.User;
import com.baomidou.mybatisplus.core.mapper.BaseMapper;
public interface UserMapper extends BaseMapper {
}
![](https://img2020.cnblogs.com/blog/1688578/202005/1688578-20200509182736101-1311032320.png)
![](https://img2020.cnblogs.com/blog/1688578/202005/1688578-20200509182747677-610854146.png)
(10)实体类、Mapper 类都写好了,就可以使用了。 Step1:先得在启动类里扫描 Mapper 类,即添加 @MapperScan 注解
package com.lyh.test.testmybatisplus;
import org.mybatis.spring.annotation.MapperScan;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
@MapperScan("mapper")
@SpringBootApplication
public class TestMybatisPlusApplication {
public static void main(String[] args) {
SpringApplication.run(TestMybatisPlusApplication.class, args);
}
}
Step2:写一个测试类测试一下。
package com.lyh.test.testmybatisplus;
import bean.User;
import mapper.UserMapper;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import java.util.List;
@SpringBootTest
class TestMybatisPlusApplicationTests {
@Autowired
private UserMapper userMapper;
@Test
public void testSelect() {
System.out.println(("----- selectAll method test ------"));
List userList = userMapper.selectList(null);
for(User user:userList) {
System.out.println(user);
}
}
}
![](https://img2020.cnblogs.com/blog/1688578/202005/1688578-20200509182838295-636931302.png)
(11)总结: 通过以上简单操作,就能对 user 表进行 CRUD 操作,不需要去编写 xml 文件。注: 若遇到 @Autowired 标记的变量出现 红色下划线,但是不影响 正常运行。
![](https://img2020.cnblogs.com/blog/1688578/202005/1688578-20200509182900276-19452208.png)
可以进入 Settings,找到 Inspection,并选择其中的 Spring Core -> Code -> Autowiring for Bean Class,将 Error 改为 Warning,即可。![](https://img2020.cnblogs.com/blog/1688578/202005/1688578-20200509182927553-240645059.png)
回到顶部
二、Mybatis-Plus 常用操作
1、配置日志
【参考地址(两种方式配置日志)】
https://blog.csdn.net/dfBeautifulLive/article/details/100700365
想要查看执行的 sql 语句,可以在 yml 文件中添加配置信息,如下。
mybatis-plus:
configuration:
log-impl: org.apache.ibatis.logging.stdout.StdOutImpl
如下图所示:执行时会打印出 sql 语句。
![](https://img2020.cnblogs.com/blog/1688578/202005/1688578-20200509183041633-571344925.png)
2、简单认识一下常用注解
【@TableName 】
@TableName 用于定义表名
注:
常用属性:
value 用于定义表名
【@TableId】
@TableId 用于定义表的主键
注:
常用属性:
value 用于定义主键字段名
type 用于定义主键类型(主键策略 IdType)
主键策略:
IdType.AUTO 主键自增,系统分配,不需要手动输入
IdType.NONE 未设置主键
IdType.INPUT 需要自己输入 主键值。
IdType.ASSIGN_ID 系统分配 ID,用于数值型数据(Long,对应 mysql 中 BIGINT 类型)。
IdType.ASSIGN_UUID 系统分配 UUID,用于字符串型数据(String,对应 mysql 中 varchar(32) 类型)。
【@TableField】
@TableField 用于定义表的非主键字段。
注:
常用属性:
value 用于定义非主键字段名
exist 用于指明是否为数据表的字段, true 表示是,false 为不是。
fill 用于指定字段填充策略(FieldFill)。
字段填充策略:(一般用于填充 创建时间、修改时间等字段)
FieldFill.DEFAULT 默认不填充
FieldFill.INSERT 插入时填充
FieldFill.UPDATE 更新时填充
FieldFill.INSERT_UPDATE 插入、更新时填充。
【@TableLogic】
@TableLogic 用于定义表的字段进行逻辑删除(非物理删除)
注:
常用属性:
value 用于定义未删除时字段的值
delval 用于定义删除时字段的值
【@Version】
@Version 用于字段实现乐观锁
3、代码生成器
(1)AutoGenerator 简介 AutoGenerator 是 MyBatis-Plus 的代码生成器,通过 AutoGenerator 可以快速生成 Entity、Mapper、Mapper XML、Service、Controller 等各个模块的代码,极大的提升了开发效率。 与 mybatis 中的 mybatis-generator-core 类似。
(2)添加依赖
com.baomidou
mybatis-plus-generator
3.3.1.tmp
org.apache.velocity
velocity-engine-core
2.2
(3)代码分析Step1: 创建一个 代码生成器。用于生成代码。 此处不用修改。
// Step1:代码生成器
AutoGenerator mpg = new AutoGenerator();
Step2: 配置全局信息。指定代码输出路径,以及包名、作者等信息。 此处按需添加,projectPath 需要修改,setAuthor 需要修改。
// Step2:全局配置
GlobalConfig gc = new GlobalConfig();
// 填写代码生成的目录(需要修改)
String projectPath = "E:\\myProject\\test\\test_mybatis_plus";
// 拼接出代码最终输出的目录
gc.setOutputDir(projectPath + "/src/main/java");
// 配置开发者信息(可选)(需要修改)
gc.setAuthor("lyh");
// 配置是否打开目录,false 为不打开(可选)
gc.setOpen(false);
// 实体属性 Swagger2 注解,添加 Swagger 依赖,开启 Swagger2 模式(可选)
//gc.setSwagger2(true);
// 重新生成文件时是否覆盖,false 表示不覆盖(可选)
gc.setFileOverride(false);
// 配置主键生成策略,此处为 ASSIGN_ID(可选)
gc.setIdType(IdType.ASSIGN_ID);
// 配置日期类型,此处为 ONLY_DATE(可选)
gc.setDateType(DateType.ONLY_DATE);
// 默认生成的 service 会有 I 前缀
gc.setServiceName("%sService");
mpg.setGlobalConfig(gc);
Step3: 配置数据源信息。用于指定 需要生成代码的 数据仓库、数据表。 setUrl、setDriverName、setUsername、setPassword 均需修改。
// Step3:数据源配置(需要修改)
DataSourceConfig dsc = new DataSourceConfig();
// 配置数据库 url 地址
dsc.setUrl("jdbc:mysql://localhost:3306/testMyBatisPlus?useUnicode=true&characterEncoding=utf8");
// dsc.setSchemaName("testMyBatisPlus"); // 可以直接在 url 中指定数据库名
// 配置数据库驱动
dsc.setDriverName("com.mysql.cj.jdbc.Driver");
// 配置数据库连接用户名
dsc.setUsername("root");
// 配置数据库连接密码
dsc.setPassword("123456");
mpg.setDataSource(dsc);
Step4: 配置包信息。 setParent、setModuleName 均需修改。其余按需求修改.
// Step:4:包配置
PackageConfig pc = new PackageConfig();
// 配置父包名(需要修改)
pc.setParent("com.lyh.test");
// 配置模块名(需要修改)
pc.setModuleName("test_mybatis_plus");
// 配置 entity 包名
pc.setEntity("entity");
// 配置 mapper 包名
pc.setMapper("mapper");
// 配置 service 包名
pc.setService("service");
// 配置 controller 包名
pc.setController("controller");
mpg.setPackageInfo(pc);
Step5: 配置数据表映射信息。 setInclude 需要修改,其余按实际开发修改。
// Step5:策略配置(数据库表配置)
StrategyConfig strategy = new StrategyConfig();
// 指定表名(可以同时操作多个表,使用 , 隔开)(需要修改)
strategy.setInclude("test_mybatis_plus_user");
// 配置数据表与实体类名之间映射的策略
strategy.setNaming(NamingStrategy.underline_to_camel);
// 配置数据表的字段与实体类的属性名之间映射的策略
strategy.setColumnNaming(NamingStrategy.underline_to_camel);
// 配置 lombok 模式
strategy.setEntityLombokModel(true);
// 配置 rest 风格的控制器(@RestController)
strategy.setRestControllerStyle(true);
// 配置驼峰转连字符
strategy.setControllerMappingHyphenStyle(true);
// 配置表前缀,生成实体时去除表前缀
// 此处的表名为 test_mybatis_plus_user,模块名为 test_mybatis_plus,去除前缀后剩下为 user。
strategy.setTablePrefix(pc.getModuleName() + "_");
mpg.setStrategy(strategy);
Step6: 执行代码生成操作。
此处不用修改。
// Step6:执行代码生成操作
mpg.execute();
完整配置如下:
package com.lyh.test.test_mybatis_plus;
import com.baomidou.mybatisplus.annotation.IdType;
import com.baomidou.mybatisplus.generator.AutoGenerator;
import com.baomidou.mybatisplus.generator.config.DataSourceConfig;
import com.baomidou.mybatisplus.generator.config.GlobalConfig;
import com.baomidou.mybatisplus.generator.config.PackageConfig;
import com.baomidou.mybatisplus.generator.config.StrategyConfig;
import com.baomidou.mybatisplus.generator.config.rules.DateType;
import com.baomidou.mybatisplus.generator.config.rules.NamingStrategy;
import org.junit.jupiter.api.Test;
public class TestAutoGenerate {
@Test
public void autoGenerate() {
// Step1:代码生成器
AutoGenerator mpg = new AutoGenerator();
// Step2:全局配置
GlobalConfig gc = new GlobalConfig();
// 填写代码生成的目录(需要修改)
String projectPath = "E:\\myProject\\test\\test_mybatis_plus";
// 拼接出代码最终输出的目录
gc.setOutputDir(projectPath + "/src/main/java");
// 配置开发者信息(可选)(需要修改)
gc.setAuthor("lyh");
// 配置是否打开目录,false 为不打开(可选)
gc.setOpen(false);
// 实体属性 Swagger2 注解,添加 Swagger 依赖,开启 Swagger2 模式(可选)
//gc.setSwagger2(true);
// 重新生成文件时是否覆盖,false 表示不覆盖(可选)
gc.setFileOverride(false);
// 配置主键生成策略,此处为 ASSIGN_ID(可选)
gc.setIdType(IdType.ASSIGN_ID);
// 配置日期类型,此处为 ONLY_DATE(可选)
gc.setDateType(DateType.ONLY_DATE);
// 默认生成的 service 会有 I 前缀
gc.setServiceName("%sService");
mpg.setGlobalConfig(gc);
// Step3:数据源配置(需要修改)
DataSourceConfig dsc = new DataSourceConfig();
// 配置数据库 url 地址
dsc.setUrl("jdbc:mysql://localhost:3306/testMyBatisPlus?useUnicode=true&characterEncoding=utf8");
// dsc.setSchemaName("testMyBatisPlus"); // 可以直接在 url 中指定数据库名
// 配置数据库驱动
dsc.setDriverName("com.mysql.cj.jdbc.Driver");
// 配置数据库连接用户名
dsc.setUsername("root");
// 配置数据库连接密码
dsc.setPassword("123456");
mpg.setDataSource(dsc);
// Step:4:包配置
PackageConfig pc = new PackageConfig();
// 配置父包名(需要修改)
pc.setParent("com.lyh.test");
// 配置模块名(需要修改)
pc.setModuleName("test_mybatis_plus");
// 配置 entity 包名
pc.setEntity("entity");
// 配置 mapper 包名
pc.setMapper("mapper");
// 配置 service 包名
pc.setService("service");
// 配置 controller 包名
pc.setController("controller");
mpg.setPackageInfo(pc);
// Step5:策略配置(数据库表配置)
StrategyConfig strategy = new StrategyConfig();
// 指定表名(可以同时操作多个表,使用 , 隔开)(需要修改)
strategy.setInclude("test_mybatis_plus_user");
// 配置数据表与实体类名之间映射的策略
strategy.setNaming(NamingStrategy.underline_to_camel);
// 配置数据表的字段与实体类的属性名之间映射的策略
strategy.setColumnNaming(NamingStrategy.underline_to_camel);
// 配置 lombok 模式
strategy.setEntityLombokModel(true);
// 配置 rest 风格的控制器(@RestController)
strategy.setRestControllerStyle(true);
// 配置驼峰转连字符
strategy.setControllerMappingHyphenStyle(true);
// 配置表前缀,生成实体时去除表前缀
// 此处的表名为 test_mybatis_plus_user,模块名为 test_mybatis_plus,去除前缀后剩下为 user。
strategy.setTablePrefix(pc.getModuleName() + "_");
mpg.setStrategy(strategy);
// Step6:执行代码生成操作
mpg.execute();
}
}
![](https://img2020.cnblogs.com/blog/1688578/202005/1688578-20200509183427652-919200133.png)
4、自动填充数据功能
(1)简介 添加、修改数据时,每次都会使用相同的方式进行填充。比如 数据的创建时间、修改时间等。 Mybatis-plus 支持自动填充这些字段的数据。
给之前的数据表新增两个字段:创建时间、修改时间。
CREATE TABLE test_mybatis_plus_user
(
id BIGINT NOT NULL COMMENT '主键ID',
name VARCHAR(30) NULL DEFAULT NULL COMMENT '姓名',
age INT(11) NULL DEFAULT NULL COMMENT '年龄',
email VARCHAR(50) NULL DEFAULT NULL COMMENT '邮箱',
create_time timestamp NULL DEFAULT NULL COMMENT '创建时间',
update_time timestamp NULL DEFAULT NULL COMMENT '最后修改时间',
PRIMARY KEY (id)
);
并使用 代码生成器生成代码。
![](https://img2020.cnblogs.com/blog/1688578/202005/1688578-20200509183526359-1836932007.png)
(2)未使用自动填充时 未使用 自动填充时,每次添加、修改数据都可以手动对其进行添加。
@SpringBootTest
class TestMybatisPlusApplicationTests {
@Autowired
private UserService userService;
@Test
public void testUpdate() {
User user = new User();
user.setName("tom").setAge(20).setEmail("[email protected]");
// 手动添加数据
user.setCreateTime(new Date()).setUpdateTime(new Date());
if (userService.save(user)) {
userService.list().forEach(System.out::println);
} else {
System.out.println("添加数据失败");
}
}
}
![](https://img2020.cnblogs.com/blog/1688578/202005/1688578-20200509183609399-44267929.png)
(3)使用自动填充功能。Step1: 使用 @TableField 注解,标注需要进行填充的字段。
/**
* 创建时间
*/
@TableField(fill = FieldFill.INSERT)
private Date createTime;
/**
* 最后修改时间
*/
@TableField(fill = FieldFill.INSERT_UPDATE)
private Date updateTime;
![](https://img2020.cnblogs.com/blog/1688578/202005/1688578-20200509183638934-543199454.png)
Step2: 自定义一个类,实现 MetaObjectHandler 接口,并重写方法。 添加 @Component 注解,交给 Spring 去管理。
package com.lyh.test.test_mybatis_plus.handler;
import com.baomidou.mybatisplus.core.handlers.MetaObjectHandler;
import org.apache.ibatis.reflection.MetaObject;
import org.springframework.stereotype.Component;
import java.util.Date;
@Component
public class MyMetaObjectHandler implements MetaObjectHandler {
@Override
public void insertFill(MetaObject metaObject) {
this.strictInsertFill(metaObject, "createTime", Date.class, new Date());
this.strictInsertFill(metaObject, "updateTime", Date.class, new Date());
}
@Override
public void updateFill(MetaObject metaObject) {
this.strictUpdateFill(metaObject, "updateTime", Date.class, new Date());
}
}
![](https://img2020.cnblogs.com/blog/1688578/202005/1688578-20200509183711767-990015180.png)
Step3: 简单测试一下。
@Test
public void testAutoFill() {
User user = new User();
user.setName("tom").setAge(20).setEmail("[email protected]");
if (userService.save(user)) {
userService.list().forEach(System.out::println);
} else {
System.out.println("添加数据失败");
}
}
![](https://img2020.cnblogs.com/blog/1688578/202005/1688578-20200509183739640-1849006710.png)
5、逻辑删除
(1)简介 删除数据,可以通过物理删除,也可以通过逻辑删除。 物理删除指的是直接将数据从数据库中删除,不保留。 逻辑删除指的是修改数据的某个字段,使其表示为已删除状态,而非删除数据,保留该数据在数据库中,但是查询时不显示该数据(查询时过滤掉该数据)。
给数据表增加一个字段:delete_flag,用于表示该数据是否被逻辑删除。
CREATE TABLE test_mybatis_plus_user
(
id BIGINT NOT NULL COMMENT '主键ID',
name VARCHAR(30) NULL DEFAULT NULL COMMENT '姓名',
age INT(11) NULL DEFAULT NULL COMMENT '年龄',
email VARCHAR(50) NULL DEFAULT NULL COMMENT '邮箱',
create_time timestamp NULL DEFAULT NULL COMMENT '创建时间',
update_time timestamp NULL DEFAULT NULL COMMENT '最后修改时间',
delete_flag tinyint(1) NULL DEFAULT NULL COMMENT '逻辑删除(0 未删除、1 删除)',
PRIMARY KEY (id)
);
(2)使用逻辑删除。 可以定义一个自动填充规则,初始值为 0。0 表示未删除, 1 表示删除。
/**
* 逻辑删除(0 未删除、1 删除)
*/
@TableLogic(value = "0", delval = "1")
@TableField(fill = FieldFill.INSERT)
private Integer deleteFlag;
@Override
public void insertFill(MetaObject metaObject) {
this.strictInsertFill(metaObject, "deleteFlag", Integer.class, 0);
}
![](https://img2020.cnblogs.com/blog/1688578/202005/1688578-20200509183843115-1369120260.png)
(3)简单测试 使用 mybatis-plus 封装好的方法时,会自动添加逻辑删除的功能。 若是自定义的 sql 语句,需要手动添加逻辑。
@Test
public void testDelete() {
if (userService.removeById(1258924257048547329L)) {
System.out.println("删除数据成功");
userService.list().forEach(System.out::println);
} else {
System.out.println("删除数据失败");
}
}
现有数据
![](https://img2020.cnblogs.com/blog/1688578/202005/1688578-20200509183959603-847557538.png)
执行 testDelete 进行逻辑删除。
![](https://img2020.cnblogs.com/blog/1688578/202005/1688578-20200509184012946-489939673.png)
![](https://img2020.cnblogs.com/blog/1688578/202005/1688578-20200509184023676-1916956821.png)
若去除 TableLogic 注解,再执行 testDelete 时进行物理删除,直接删除这条数据。
![](https://img2020.cnblogs.com/blog/1688578/202005/1688578-20200509184044446-583363399.png)
![](https://img2020.cnblogs.com/blog/1688578/202005/1688578-20200509184057341-414839563.png)
6、分页插件的使用
(1)简介 与 mybatis 的插件 pagehelper 用法类似。 通过简单的配置即可使用。
(2)使用Step1: 配置分页插件。 编写一个 配置类,内部使用 @Bean 注解将 PaginationInterceptor 交给 Spring 容器管理。
package com.lyh.test.test_mybatis_plus.config;
import com.baomidou.mybatisplus.extension.plugins.PaginationInterceptor;
import org.mybatis.spring.annotation.MapperScan;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
/**
* 自定义一个配置类,mapper 扫描也可在此写上
*/
@Configuration
@MapperScan("com.lyh.test.test_mybatis_plus.mapper")
public class Myconfig {
/**
* 分页插件
* @return 分页插件的实例
*/
@Bean
public PaginationInterceptor paginationInterceptor() {
return new PaginationInterceptor();
}
}
![](https://img2020.cnblogs.com/blog/1688578/202005/1688578-20200509184134459-1998710648.png)
Step2: 编写分页代码。 直接 new 一个 Page 对象,对象需要传递两个参数(当前页,每页显示的条数)。 调用 mybatis-plus 提供的分页查询方法,其会将 分页查询的数据封装到 Page 对象中。
@Test
public void testPage() {
// Step1:创建一个 Page 对象
Page page = new Page();
// Page page = new Page(2, 5);
// Step2:调用 mybatis-plus 提供的分页查询方法
userService.page(page, null);
// Step3:获取分页数据
System.out.println(page.getCurrent()); // 获取当前页
System.out.println(page.getTotal()); // 获取总记录数
System.out.println(page.getSize()); // 获取每页的条数
System.out.println(page.getRecords()); // 获取每页数据的集合
System.out.println(page.getPages()); // 获取总页数
System.out.println(page.hasNext()); // 是否存在下一页
System.out.println(page.hasPrevious()); // 是否存在上一页
}
![](https://img2020.cnblogs.com/blog/1688578/202005/1688578-20200509184212763-1123404604.png)
7、乐观锁的实现
(1)首先认识一下 读问题、写问题? 操作数据库数据时,遇到的最基本问题就是 读问题与写问题。 读问题 指的是从数据库中读取数据时遇到的问题,比如:脏读、幻读、不可重复读。
【脏读、幻读、不可重复读 参考地址:】
https://www.cnblogs.com/l-y-h/p/12458777.html#_label0_3
写问题 指的是数据写入数据库时遇到的问题,比如:丢失更新(多个线程同时对某条数据更新,无论执行顺序如何,都会丢失其他线程更新的数据)
(2)如何解决写问题? 乐观锁、悲观锁就是为了解决 写问题而存在的。 乐观锁:总是假设最好的情况,每次读取数据时认为数据不会被修改(即不加锁),当进行更新操作时,会判断这条数据是否被修改,未被修改,则进行更新操作。若被修改,则数据更新失败,可以对数据进行重试(重新尝试修改数据)。 悲观锁:总是假设最坏的情况,每次读取数据时认为数据会被修改(即加锁),当进行更新操作时,直接更新数据,结束操作后释放锁(此处才可以被其他线程读取)。
(3)乐观锁、悲观锁使用场景? 乐观锁一般用于读比较多的场合,尽量减少加锁的开销。 悲观锁一般用于写比较多的场合,尽量减少 类似 乐观锁重试更新引起的性能开销。
(4)乐观锁两种实现方式方式一:通过版本号机制实现。 在数据表中增加一个 version 字段。 取数据时,获取该字段,更新时以该字段为条件进行处理(即set version = newVersion where version = oldVersion),若 version 相同,则更新成功(给新 version 赋一个值,一般加 1)。若 version 不同,则更新失败,可以重新尝试更新操作。
方式二:通过 CAS 算法实现。 CAS 为 Compare And Swap 的缩写,即比较交换,是一种无锁算法(即在不加锁的情况实现多线程之间的变量同步)。 CAS 操作包含三个操作数 —— 内存值(V)、预期原值(A)和新值(B)。如果内存地址里面的值 V 和 A 的值是一样的,那么就将内存里面的值更新成B。若 V 与 A 不一致,则不执行任何操作(可以通过自旋操作,不断尝试修改数据直至成功修改)。即 V == A ? V = B : V = V。 CAS 可能导致 ABA 问题(两次读取数据时值相同,但不确定值是否被修改过),比如两个线程操作同一个变量,线程 A、线程B 初始读取数据均为 A,后来 线程B 将数据修改为 B,然后又修改为 A,此时线程 A 再次读取到的数据依旧是 A,虽然值相同但是中间被修改过,这就是 ABA 问题。可以加一个额外的标志位 C,用于表示数据是否被修改。当标志位 C 与预期标志位相同、且 V == A 时,则更新值 B。
(5)mybatis-plus 实现乐观锁(通过 version 机制)实现思路: Step1:取出记录时,获取当前version Step2:更新时,带上这个version Step3:执行更新时, set version = newVersion where version = oldVersion Step4:如果version不对,就更新失败
(6)mybatis-plus 代码实现乐观锁Step1: 配置乐观锁插件。 编写一个配置类(可以与上例的分页插件共用一个配置类),将 OptimisticLockerInterceptor 通过 @Bean 交给 Spring 管理。
/**
* 乐观锁插件
* @return 乐观锁插件的实例
*/
@Bean
public OptimisticLockerInterceptor optimisticLockerInterceptor() {
return new OptimisticLockerInterceptor();
}
![](https://img2020.cnblogs.com/blog/1688578/202005/1688578-20200509184345369-1234458679.png)
Step2: 定义一个数据库字段 version。
CREATE TABLE test_mybatis_plus_user
(
id BIGINT NOT NULL COMMENT '主键ID',
name VARCHAR(30) NULL DEFAULT NULL COMMENT '姓名',
age INT(11) NULL DEFAULT NULL COMMENT '年龄',
email VARCHAR(50) NULL DEFAULT NULL COMMENT '邮箱',
create_time timestamp NULL DEFAULT NULL COMMENT '创建时间',
update_time timestamp NULL DEFAULT NULL COMMENT '最后修改时间',
delete_flag tinyint(1) NULL DEFAULT NULL COMMENT '逻辑删除(0 未删除、1 删除)',
version int NULL DEFAULT NULL COMMENT '版本号(用于乐观锁, 默认为 1)',
PRIMARY KEY (id)
);
Step3: 使用 @Version 注解标注对应的实体类。 可以通过 @TableField 进行数据自动填充。
/**
* 版本号(用于乐观锁, 默认为 1)
*/
@Version
@TableField(fill = FieldFill.INSERT)
private Integer version;
@Override
public void insertFill(MetaObject metaObject) {
this.strictInsertFill(metaObject, "version", Integer.class, 1);
}
![](https://img2020.cnblogs.com/blog/1688578/202005/1688578-20200509184439393-659580638.png)
Step4: 简单测试一下,可以看一下 控制台 打印的 sql 语句。
@Test
public void testVersion() {
User user = new User();
user.setName("tom").setAge(20).setEmail("[email protected]");
userService.save(user);
userService.list().forEach(System.out::println);
user.setName("jarry");
userService.update(user, null);
userService.list().forEach(System.out::println);
}
![](https://img2020.cnblogs.com/blog/1688578/202005/1688578-20200509184519097-1477211602.png)
回到顶部
三、Mybatis-Plus CRUD 操作简单了解一下
1、Mapper 接口方法(CRUD)简单了解一下
(1)简介 使用代码生成器生成的 mapper 接口中,其继承了 BaseMapper 接口。 而 BaseMapper 接口中封装了一系列 CRUD 常用操作,可以直接使用,而不用自定义 xml 与 sql 语句进行 CRUD 操作(当然根据实际开发需要,自定义 sql 还是有必要的)。
此处简单介绍一下 BaseMapper 接口中的常用方法。
![](https://img2020.cnblogs.com/blog/1688578/202005/1688578-20200509184606042-2140497864.png)
![](https://img2020.cnblogs.com/blog/1688578/202005/1688578-20200509184616508-1167828998.png)
(2)方法介绍 混个眼熟,用多了就记得了。
【添加数据:(增)】
int insert(T entity); // 插入一条记录
注:
T 表示任意实体类型
entity 表示实体对象
【删除数据:(删)】
int deleteById(Serializable id); // 根据主键 ID 删除
int deleteByMap(@Param(Constants.COLUMN_MAP) Map columnMap); // 根据 map 定义字段的条件删除
int delete(@Param(Constants.WRAPPER) Wrapper wrapper); // 根据实体类定义的 条件删除对象
int deleteBatchIds(@Param(Constants.COLLECTION) Collection |